import { colord, extend } from 'colord';
import mixPlugin from 'colord/plugins/mix';
import type { HsvColor } from 'colord';

extend([mixPlugin]);

type ColorIndex = 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10;

const hueStep = 2;
const saturationStep = 16;
const saturationStep2 = 5;
const brightnessStep1 = 5;
const brightnessStep2 = 15;
const lightColorCount = 5;
const darkColorCount = 4;

/**
 * 根据颜色获取调色板颜色(从左至右颜色从浅到深，6为主色号)
 * @param color - 颜色
 * @param index - 调色板的对应的色号(6为主色号)
 * @description 算法实现从ant-design调色板算法中借鉴 https://github.com/ant-design/ant-design/blob/master/components/style/color/colorPalette.less
 */
export function getColorPalette(color: string, index: ColorIndex) {
	if (index === 6) return color;

	const isLight = index < 6;
	const hsv = colord(color).toHsv();
	const i = isLight ? lightColorCount + 1 - index : index - lightColorCount - 1;

	const newHsv = {
		h: getHue(hsv, i, isLight),
		s: getSaturation(hsv, i, isLight),
		v: getValue(hsv, i, isLight),
	};

	return colord(newHsv).toHex();
}

/**
 * 根据颜色获取调色板颜色所有颜色
 * @param color - 颜色
 */
export function getAllColorPalette(color: string) {
	const indexs: ColorIndex[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
	return indexs.map((index) => getColorPalette(color, index));
}

/**
 * 获取色相渐变
 * @param hsv - hsv格式颜色值
 * @param i - 与6的相对距离
 * @param isLight - 是否是亮颜色
 */
function getHue(hsv: any, i: number, isLight: boolean) {
	let hue: number;
	if (hsv.h >= 60 && hsv.h <= 240) {
		// 冷色调
		// 减淡变亮 色相顺时针旋转 更暖
		// 加深变暗 色相逆时针旋转 更冷
		hue = isLight ? hsv.h - hueStep * i : hsv.h + hueStep * i;
	} else {
		// 暖色调
		// 减淡变亮 色相逆时针旋转 更暖
		// 加深变暗 色相顺时针旋转 更冷
		hue = isLight ? hsv.h + hueStep * i : hsv.h - hueStep * i;
	}
	if (hue < 0) {
		hue += 360;
	} else if (hue >= 360) {
		hue -= 360;
	}
	return hue;
}

/**
 * 获取饱和度渐变
 * @param hsv - hsv格式颜色值
 * @param i - 与6的相对距离
 * @param isLight - 是否是亮颜色
 */
function getSaturation(hsv: any, i: number, isLight: boolean) {
	let saturation: number;
	if (isLight) {
		saturation = hsv.s - saturationStep * i;
	} else if (i === darkColorCount) {
		saturation = hsv.s + saturationStep;
	} else {
		saturation = hsv.s + saturationStep2 * i;
	}
	if (saturation > 100) {
		saturation = 100;
	}
	if (isLight && i === lightColorCount && saturation > 10) {
		saturation = 10;
	}
	if (saturation < 6) {
		saturation = 6;
	}
	return saturation;
}

/**
 * 获取明度渐变
 * @param hsv - hsv格式颜色值
 * @param i - 与6的相对距离
 * @param isLight - 是否是亮颜色
 */
function getValue(hsv: any, i: number, isLight: boolean) {
	let value: number;
	if (isLight) {
		value = hsv.v + brightnessStep1 * i;
	} else {
		value = hsv.v - brightnessStep2 * i;
	}
	if (value > 100) {
		value = 100;
	}
	return value;
}

/**
 * 给颜色加透明度
 * @param color - 颜色
 * @param alpha - 透明度(0 - 1)
 */
export function addColorAlpha(color: string, alpha: number) {
	return colord(color).alpha(alpha).toHex();
}

/**
 * 颜色混合
 * @param firstColor - 第一个颜色
 * @param secondColor - 第二个颜色
 * @param ratio - 第二个颜色占比
 */
export function mixColor(firstColor: string, secondColor: string, ratio: number) {
	return colord(firstColor).mix(secondColor, ratio).toHex();
}

/**
 * 是否是白颜色
 * @param color - 颜色
 */
export function isWhiteColor(color: string) {
	return colord(color).isEqual('#ffffff');
}
